/* eslint-disable max-lines-per-function */
import { test, expect, type Page } from '@playwright/test';
import { addCoverage } from '../../playwright-coverage';
import type { IRouter } from '@aurelia/router-direct';

test.describe('router', () => {

  addCoverage();

  test.describe.configure({ mode: 'parallel' });

  test.beforeEach(async ({ page, baseURL }) => {
    await page.goto(baseURL);
  });

  test('loads home route', async ({ page }) => {
    await expect(page.locator('#root-vp')).toHaveText('Home page');
  });

  test('reloads home route', async ({ page }) => {
    await expect(page.locator('#root-vp')).toHaveText('Home page');
    await page.reload();
    await expect(page.locator('#root-vp')).toHaveText('Home page');
  });

  test('loads route when clicking on link', async ({ page }) => {
    await page.click('a:text("Register")');
    await expect(page.locator('#root-vp')).toContainText('Auth page');
  });

  test('loads right component refreshing the page with deep linking', async ({ page, baseURL }) => {
    await page.click(':text("Show iframe")');
    const frame = await (await page.$('iframe')).contentFrame();
    await Promise.all([
      page.waitForNavigation(),
      frame.click(':text("Goto auth")'),
    ]);
    expect(page.url()).toBe(`${baseURL}/auth`);
    await expect(page.locator('#root-vp')).toContainText('Auth page');
  });

  test('loads right component refreshing the page with deep linking - url fragment hash', async ({ page, baseURL }) => {
    await page.click(':text("Toggle url fragment hash")');

    await page.click(':text("Show iframe")');
    const frame = await (await page.$('iframe')).contentFrame();
    await Promise.all([
      page.waitForNavigation(),
      frame.click(':text("Goto auth")'),
    ]);
    expect(page.url()).toBe(`${baseURL}/#/auth`);
    await expect(page.locator('#root-vp')).toContainText('Auth page');
  });

  test('loads fallback component when clicking on link with missing component', async ({ page }) => {
    await page.click('a:text("Something missing")');
    await expect(page.locator('#root-vp')).toContainText('Fallback for: something/missing');
  });

  test('loads fallback component when clicking on link in child with missing component', async ({ page }) => {
    await page.click('a:text("Child")');
    await page.click('a:text("Something missing in child")');
    await expect(page.locator('#child-vp')).toContainText('Fallback for: something/missing');
  });

  const tests = [
    { link: 'Pages one-route', result: 'One page' },
    { link: 'Pages two-route', result: 'Two page' },
  ];

  for (const t of tests) {
    test(`loads async route when clicking on link "${t.link}"`, async ({ page }) => {
      await page.click(`a:text("${t.link}")`);
      await expect(page.locator('#root-vp')).toContainText(t.result);
    });
  }

  test('sets the right href based on load attribute', async ({ page }) => {
    await page.click('a:text("Child")');
    await expect(page.locator('#-link')).toHaveAttribute('href', '');
    await expect(page.locator('#auth-link')).toHaveAttribute('href', 'auth');
    await expect(page.locator('#child-auth-link')).toHaveAttribute('href', 'auth');
    await expect(page.locator('#child-something-missing-link')).toHaveAttribute('href', 'child/something/missing');
  });

  test('goes from home to auth page, then go back', async ({ page, baseURL }) => {
    // Home
    await expect(page.locator('#root-vp')).toHaveText('Home page');

    // Go to Auth
    await Promise.all([
      page.waitForURL(`${baseURL}/auth`),
      page.click('#auth-link'),
    ]);
    expect(page.url()).toBe(`${baseURL}/auth`);
    await expect(page.locator('#root-vp')).toContainText('Auth page');

    // Go to Home using the 'home' url
    await Promise.all([
      page.waitForURL(`${baseURL}/`),
      page.click('#-link'),
    ]);
    expect(page.url()).toBe(`${baseURL}/`);
    await expect(page.locator('#root-vp')).toHaveText('Home page');

    // Go to Auth again
    await Promise.all([
      page.waitForURL(`${baseURL}/auth`),
      page.click('#auth-link'),
    ]);
    expect(page.url()).toBe(`${baseURL}/auth`);
    await expect(page.locator('#root-vp')).toContainText('Auth page');

    // Got to Home using the go back
    await Promise.all([
      page.waitForURL(`${baseURL}/`),
      page.goBack(),
    ]);

    expect(page.url()).toBe(`${baseURL}/`);
    await expect(page.locator('#root-vp')).toHaveText('Home page');
  });

  test('goes from home to component page, then go back', async ({ page, baseURL }) => {
    // Home
    await expect(page.locator('#root-vp')).toHaveText('Home page');

    // Go to "one"
    await Promise.all([
      page.waitForURL(`${baseURL}/pages/one-route`),
      page.click('#page-one-link'),
    ]);
    expect(page.url()).toBe(`${baseURL}/pages/one-route`);
    await expect(page.locator('#root-vp')).toContainText('One page');

    // Go to "two"
    await Promise.all([
      page.waitForURL(`${baseURL}/pages/two-route`),
      page.click('#page-two-link'),
    ]);
    expect(page.url()).toBe(`${baseURL}/pages/two-route`);
    await expect(page.locator('#root-vp')).toContainText('Two page');

    // Go back to "one"
    await Promise.all([
      page.waitForURL(`${baseURL}/pages/one-route`),
      page.goBack(),
    ]);
    expect(page.url()).toBe(`${baseURL}/pages/one-route`);
    await expect(page.locator('#root-vp')).toContainText('One page');

    // Go back to "home"
    await Promise.all([
      page.waitForURL(`${baseURL}/`),
      page.goBack(),
    ]);
    expect(page.url()).toBe(`${baseURL}/`);
    await expect(page.locator('#root-vp')).toContainText('Home page');
  });

  // with the loop, the issue with duplicate content is much easier to reproduce
  for (let i = 0; i < 5; i++) {
    test(`ensures no duplication with load${i}`, async ({ page, baseURL }) => {
      // Home
      const proms: Promise<void>[] = [];
      await expect(page.locator('#root-vp')).toHaveText('Home page');
      // Flood navigation
      for (let i = 0; i < 50; i++) {
        switch (i % 3) {
          case 0:
            proms.push(page.click('#page-one-link'));
            break;
          case 1:
            proms.push(page.click('#page-two-link'));
            break;
          case 2:
            proms.push(page.click('#auth-link'));
            break;
        }
      }

      await Promise.all(proms);

      await _waitProcessingNav(page);

      // Go to "two"
      await Promise.all([
        page.waitForURL(`${baseURL}/pages/two-route`),
        page.click('#page-two-link'),
      ]);
      expect(page.url()).toBe(`${baseURL}/pages/two-route`);
      expect(await page.locator('#root-vp').textContent()).toBe('Two page');
    });
  }

  test('ensures no duplication with back/forward', async ({ page, baseURL }) => {
    // Home
    await expect(page.locator('#root-vp')).toHaveText('Home page');

    // Go to "one"
    await Promise.all([
      page.waitForURL(`${baseURL}/pages/one-route`),
      page.click('#page-one-link'),
    ]);
    expect(page.url()).toBe(`${baseURL}/pages/one-route`);
    await expect(page.locator('#root-vp')).toContainText('One page');

    // Go to "two"
    await Promise.all([
      page.waitForURL(`${baseURL}/pages/two-route`),
      page.click('#page-two-link'),
    ]);

    // Flood back/forward
    const proms: Promise<unknown>[] = [];
    for (let i = 0; i < 98; i++) {
      if (i % 2 === 0) {
        proms.push(page.goBack(), page.goBack());
      } else {
        proms.push(page.goForward(), page.goForward());
      }
    }
    await Promise.all(proms);

    await _waitProcessingNav(page);

    // Go to "two"
    await Promise.all([
      page.waitForURL(`${baseURL}/pages/two-route`),
      page.click('#page-two-link'),
    ]);

    expect(page.url()).toBe(`${baseURL}/pages/two-route`);
    expect(await page.locator('#root-vp').textContent()).toBe('Two page');
  });

  test('loads route with parameters', async ({ page }) => {
    await page.click('a:text("Pages params-route 123")');
    await expect(page.locator('#root-vp')).toContainText('Params page');
    await expect(page.locator('#params')).toHaveText('{"p":"123"}');
  });

  test('reloads route with parameters', async ({ page }) => {
    await page.click('a:text("Pages params-route 123")');
    await expect(page.locator('#root-vp')).toContainText('Params page');
    await expect(page.locator('#params')).toHaveText('{"p":"123"}');
    await page.reload();
    await expect(page.locator('#root-vp')).toContainText('Params page');
    await expect(page.locator('#params')).toHaveText('{"p":"123"}');
  });
});

/** Wait until the navigation processing is complete */
async function _waitProcessingNav(page: Page): Promise<void> {
  let isNav: boolean;
  do {
    // eslint-disable-next-line no-await-in-loop
    await new Promise(resolve => setTimeout(resolve, 200));
    // eslint-disable-next-line no-await-in-loop
    isNav = await page.evaluate(() => (window as Window & { _auRouter?: IRouter & { isProcessingNav?: boolean } })._auRouter.isProcessingNav);
  } while (isNav);
}
